Improve explicit drawing and flushing (fixes GtkRuler)
authorRichard Hult <richard@imendio.com>
Wed, 18 Feb 2009 15:20:06 +0000 (16:20 +0100)
committerAlexander Larsson <alex@localhost.localdomain>
Thu, 2 Apr 2009 08:16:30 +0000 (10:16 +0200)
Use the same code path to get a CGContext for both gdk_draw_* and
gdk_cairo_create and make sure we unlockFocus in both cases. This
fixes the broken rendering in GtkRuler. Also use an average of flush
intervals when checking whether we can flush or not, since otherwise
we get too sensitive and block almost all explicit flushes that are
caused by mouse movements for example.

gdk/quartz/gdkdrawable-quartz.c
gdk/quartz/gdkprivate-quartz.h
gdk/quartz/gdkwindow-quartz.c

index 97b17a97290938f57cee3f230051429945e76e80..72b678c0c5636d41bce5215905c3a73e53b6f296 100644 (file)
@@ -37,7 +37,21 @@ _gdk_windowing_set_cairo_surface_size (cairo_surface_t *surface,
                                       int              width,
                                       int              height)
 {
-  /* FIXME: we must recreate the surface here. */
+  /* This is not supported with quartz surfaces. */
+}
+
+static void
+gdk_quartz_cairo_surface_destroy (void *data)
+{
+  GdkQuartzCairoSurfaceData *surface_data = data;
+  GdkDrawableImplQuartz *impl = GDK_DRAWABLE_IMPL_QUARTZ (surface_data->drawable);
+
+  impl->cairo_surface = NULL;
+
+  gdk_quartz_drawable_release_context (surface_data->drawable,
+                                       surface_data->cg_context);
+
+  g_free (surface_data);
 }
 
 cairo_surface_t *
@@ -45,33 +59,27 @@ _gdk_windowing_create_cairo_surface (GdkDrawable *drawable,
                                     int          width,
                                     int          height)
 {
-      CGContextRef cg_context;
-      cairo_surface_t *surface;
+  CGContextRef cg_context;
+  GdkQuartzCairoSurfaceData *surface_data;
+  cairo_surface_t *surface;
 
-      /* FIXME: Can this ever be called on a non-window drawable? If so we
-       * need to check whether it's a window or pixmap and get the right
-       * kind of context, and also release it in destroy below (for bitmap
-       * context).
-       */
+  cg_context = gdk_quartz_drawable_get_context (drawable, TRUE);
 
-      cg_context = [[NSGraphicsContext currentContext] graphicsPort];
-      if (!cg_context)
-       return NULL;
+  if (!cg_context)
+    return NULL;
 
-      surface = cairo_quartz_surface_create_for_cg_context (cg_context, width, height);
+  surface_data = g_new (GdkQuartzCairoSurfaceData, 1);
+  surface_data->drawable = drawable;
+  surface_data->cg_context = cg_context;
 
-      return surface;
-}
+  surface = cairo_quartz_surface_create_for_cg_context (cg_context,
+                                                        width, height);
 
-static void
-gdk_quartz_cairo_surface_destroy (void *data)
-{
-  GdkQuartzCairoSurfaceData *surface_data = data;
-  GdkDrawableImplQuartz *impl = GDK_DRAWABLE_IMPL_QUARTZ (surface_data->drawable);
+  cairo_surface_set_user_data (surface, &gdk_quartz_cairo_key,
+                               surface_data,
+                               gdk_quartz_cairo_surface_destroy);
 
-  impl->cairo_surface = NULL;
-
-  g_free (surface_data);
+  return surface;
 }
 
 static cairo_surface_t *
@@ -88,20 +96,8 @@ gdk_quartz_ref_cairo_surface (GdkDrawable *drawable)
       int width, height;
 
       gdk_drawable_get_size (drawable, &width, &height);
-
-      impl->cairo_surface = _gdk_windowing_create_cairo_surface (drawable, width, height);
-
-      if (impl->cairo_surface)
-        {
-          GdkQuartzCairoSurfaceData *surface_data;
-
-          surface_data = g_new (GdkQuartzCairoSurfaceData, 1);
-          surface_data->drawable = drawable;
-          surface_data->cg_context = cairo_quartz_surface_get_cg_context (impl->cairo_surface);
-
-          cairo_surface_set_user_data (impl->cairo_surface, &gdk_quartz_cairo_key,
-                                       surface_data, gdk_quartz_cairo_surface_destroy);
-        }
+      impl->cairo_surface = _gdk_windowing_create_cairo_surface (drawable,
+                                                                 width, height);
     }
   else
     cairo_surface_reference (impl->cairo_surface);
@@ -381,6 +377,7 @@ gdk_quartz_draw_drawable (GdkDrawable *drawable,
     }
   else if (dest_depth != 0 && src_depth == dest_depth)
     {
+      GdkPixmapImplQuartz *pixmap_impl = GDK_PIXMAP_IMPL_QUARTZ (src_impl);
       CGContextRef context = gdk_quartz_drawable_get_context (drawable, FALSE);
 
       if (!context)
@@ -391,14 +388,12 @@ gdk_quartz_draw_drawable (GdkDrawable *drawable,
 
       CGContextClipToRect (context, CGRectMake (xdest, ydest, width, height));
       CGContextTranslateCTM (context, xdest - xsrc, ydest - ysrc +
-                             GDK_PIXMAP_IMPL_QUARTZ (src_impl)->height);
+                             pixmap_impl->height);
       CGContextScaleCTM (context, 1.0, -1.0);
 
       CGContextDrawImage (context,
-                          CGRectMake(0, 0,
-                                     GDK_PIXMAP_IMPL_QUARTZ (src_impl)->width,
-                                     GDK_PIXMAP_IMPL_QUARTZ (src_impl)->height),
-                          GDK_PIXMAP_IMPL_QUARTZ (src_impl)->image);
+                          CGRectMake (0, 0, pixmap_impl->width, pixmap_impl->height),
+                          pixmap_impl->image);
 
       gdk_quartz_drawable_release_context (drawable, context);
     }
@@ -726,32 +721,44 @@ gdk_quartz_drawable_get_context (GdkDrawable *drawable,
   return GDK_DRAWABLE_IMPL_QUARTZ_GET_CLASS (drawable)->get_context (drawable, antialias);
 }
 
-/* Help preventing "beam synch penalty" where CG makes all graphics code
+/* Help preventing "beam sync penalty" where CG makes all graphics code
  * block until the next vsync if we try to flush (including call display on
  * a view) too often. We do this by limiting the manual flushing done
- * outside of expose calls to less than 20Hz, this should leave enough room
- * for the 60Hz max rate including the "regular" flushing.
+ * outside of expose calls to less than some frequency when measured over
+ * the last 4 flushes. This is a bit arbitray, but seems to make it possible
+ * for some quick manual flushes (such as gtkruler or gimp's marching ants)
+ * without hitting the max flush frequency.
  *
- * If cg_context is NULL, no flushing is done, only registering that a flush
- * was made externally.
+ * If drawable NULL, no flushing is done, only registering that a flush was
+ * done externally.
  */
 void
-_gdk_quartz_drawable_flush (CGContextRef cg_context)
+_gdk_quartz_drawable_flush (GdkDrawable *drawable)
 {
   static struct timeval prev_tv;
+  static gint intervals[4];
+  static gint index;
   struct timeval tv;
   gint ms;
 
   gettimeofday (&tv, NULL);
+  ms = (tv.tv_sec - prev_tv.tv_sec) * 1000 + (tv.tv_usec - prev_tv.tv_usec) / 1000;
+  intervals[index++ % 4] = ms;
 
-  if (cg_context)
+  if (drawable)
     {
-      ms = (tv.tv_sec - prev_tv.tv_sec) * 1000 + (tv.tv_usec - prev_tv.tv_usec) / 1000;
+      ms = intervals[0] + intervals[1] + intervals[2] + intervals[3];
 
-      /* ~20Hz. */
-      if (ms > 50)
+      /* ~25Hz on average. */
+      if (ms > 4*40)
         {
-          CGContextFlush (cg_context);
+          if (GDK_IS_WINDOW_IMPL_QUARTZ (drawable))
+            {
+              GdkWindowImplQuartz *window_impl = GDK_WINDOW_IMPL_QUARTZ (drawable);
+
+              [window_impl->toplevel flushWindow];
+            }
+
           prev_tv = tv;
         }
     }
@@ -773,7 +780,7 @@ gdk_quartz_drawable_release_context (GdkDrawable  *drawable,
       /* See comment in gdk_quartz_drawable_get_context(). */
       if (window_impl->in_paint_rect_count == 0)
         {
-          _gdk_quartz_drawable_flush (cg_context);
+          _gdk_quartz_drawable_flush (drawable);
           [window_impl->view unlockFocus];
         }
     }
index 8bb7ea596ae50359ca2eca75396ad0ff50155fc5..9590d9b561d115e50692ec6a19bfaa7a5abe28fe 100644 (file)
@@ -180,7 +180,7 @@ gboolean     _gdk_quartz_keys_is_modifier (guint      keycode);
 
 /* Drawable */
 void        _gdk_quartz_drawable_finish (GdkDrawable *drawable);
-void        _gdk_quartz_drawable_flush  (CGContextRef cg_context);
+void        _gdk_quartz_drawable_flush  (GdkDrawable *drawable);
 
 /* Geometry */
 void        _gdk_quartz_window_scroll      (GdkWindow       *window,
index 2bb4bdb1c32e532f169ebcf0a8eb8ab5245f80e0..aa8d437a4684d0d055fbf0033e164a8ab71fcd35 100644 (file)
@@ -92,7 +92,7 @@ gdk_window_impl_quartz_get_context (GdkDrawable *drawable,
    * is needed when called from outside "real" expose events, for
    * example for synthesized expose events when realizing windows
    * and for widgets that send fake expose events like the arrow
-   * buttons in spinbuttons.
+   * buttons in spinbuttons or the position marker in rulers.
    */
   if (window_impl->in_paint_rect_count == 0)
     {